\chapter{自定义 \LaTeX{} 命令和功能}\label{chap:custom}
\addtocontents{los}{\protect\addvspace{10pt}}

\begin{intro}
读到这一章之前，如果你确保掌握了前几章的知识并熟练运用，你已经能制作出内容和形式都相当丰富的文档了。
但你可能还不会满足：我要如何制作一个简单但像样的毕业论文/书籍/简历模板，每次可以直接套用，
而不是再在导言区写一堆代码？

本章的内容将有助于你实现这一个目标，让你能编写可重复利用的模块——宏包和文档类，并在其中自己定义命令和环境。
不过作为入门手册，这些知识仍是不全面的。如果你不满足于此，需要参考更多资料，比如 \cite{companion,clsguide}。
\end{intro}

\section{自定义命令和环境}\label{sec:defs}

你也许已经意识到了，在本手册中介绍的所有命令都被包含在一个带颜色的矩形背景框中。
笔者并没有直接使用基础的 \LaTeX{} 命令来实现这个效果，而是创建了一个\textbf{宏包}，
并在其中定义了笔者所需要的命令和环境。现在我只需写成这样简单的形式：

\begin{example}
\begin{command}
\cmd{dum}
\end{command}
\end{example}

这个例子中使用了一个新的环境 \env{command}。这个环境负责给命令代码加上一个带颜色的矩形背景框。
同时还使用了一个命令：\cmd{cmd}， 这个命令负责输出命令的名字，包括前面的反斜线。

一旦笔者想要修改命令代码的样式，比如更换颜色、加边框等等，可以通过改变 \env{command} 环境的定义来很容易地创建新的外观，
而不是挨个修改每个命令示例。

\subsection{定义新命令}\label{subsec:newcmd}

\cmdindex{newcommand}
使用如下命令可以定义你自己的命令：
\begin{command}
\cmd{newcommand}\marg*{\cmd{\Arg{name}}}\oarg{num}\marg{definition}
\end{command}
\cmd{newcommand} 的基本用法需要两个必选参数，第一个参数 \Arg{name} 是要定义的命令名称（带反斜线），
第二个参数 \Arg{definition} 是命令的具体定义。方括号里的参数 \Arg{num} 是可选的，
用于指定新命令所需的参数数目（最多 9 个）。如果缺省可选参数，默认就是 0，也就是新定义的命令不带任何参数。

接下来的两个例子有助于理解。第一个例子定义了一个新的命令 \cmd{tnss}。
这个命令是本手册英文名称“The Not So Short Introduction to \LaTeXe” 的简写。
如果需要在文档中多次使用本手册的名称，那么使用这个命令是一个非常方便的办法。

\begin{example}
\newcommand{\tnss}{The not so Short
  Introduction to \LaTeXe}
This is ``\tnss'' \ldots{} ``\tnss''
\end{example}

第二个例子演示了如何定义一个带参数的命令。在命令的定义中，标记 \verb|#1|
代表指定的参数。如果想使用多个参数，可以依次使用 \verb|#2|、……、\verb|#9| 等标记。

\begin{example}
\newcommand{\txsit}[1]{This is the
  \emph{#1} Short Introduction
  to \LaTeXe}
% in the document body:
\begin{itemize}
\item \txsit{not so}
\item \txsit{very}
\end{itemize}
\end{example}

\cmdindex{renewcommand}
\LaTeX{} 不允许使用 \cmd{newcommand} 定义一个与现有命令重名的命令。如果需要修改命令定义的话，
使用 \cmd{re\-new\-comm\-and} 命令。它使用与命令 \cmd{new\-comm\-and} 相同的语法。

\cmdindex{providecommand}
在某些情况之下，使用 \cmd{providecommand} 命令是一种比较理想的方案：在命令未定义时，
它相当于 \cmd{new\-comm\-and}；在命令已定义时，沿用已有的定义。

\subsection{定义环境}\label{subsec:newenv}

\cmdindex{newenvironment}
与 \cmd{new\-comm\-and} 命令类似，可以用 \cmd{new\-environ\-ment} 定义新的环境。它的语法如下所示：

\begin{command}
\cmd{newenvironment}\marg{name}\oarg{num}\marg{before}\marg{after}
\end{command}

同样地，\cmd{newenvironment} 命令有一个可选的参数。在 \Arg{before} 中的内容将在此环境包含
的文本之前处理，而在 \Arg{after} 中的内容将在遇到 \cmd{end\marg{name}} 命令时处理。

下面的例子演示了 \cmd{newenvironment} 命令的用法：
\begin{example}
\newenvironment{king}
{\rule{1ex}{1ex}%
     \hspace{\stretch{1}}}
{\hspace{\stretch{1}}%
     \rule{1ex}{1ex}}

\begin{king}
My humble subjects \ldots
\end{king}
\end{example}

\cmdindex{renewenvironment}
参数 \Arg{num} 的使用方式与 \cmd{new\-comm\-and} 命令相同。\LaTeX{} 还同样保证你
不会不小心新建重名的环境。如果你确实希望改变一个现有的环境，你可以使用命令
 \cmd{re\-new\-environ\-ment}，它使用和命令 \cmd{new\-environ\-ment} 相同的语法。

\subsection{\pkg{xparse} 宏包简介}\label{subsec:xparse}

\pkgindex{xparse}
\cmdindex[xparse]{NewDocumentCommand,NewDocumentEnvironment}
通过 \cmd{newcommand} 和 \cmd{newenvironment} 定义的命令或环境格式比较固定。如果需要定义带有多个
可选参数、或者带星号的命令或环境，可以使用 \pkg{xparse} 宏包%
\footnote{\LaTeX{} 2020-10-01 版本之后，\pkg{xparse} 宏包已集成在了格式之中，不需要显式调用。}。
它提供了 \cmd{New\-Document\-Command} 和 \cmd{New\-Document\-Environment} 等命令，具体语法如下：

\begin{command}
\cmd{NewDocumentCommand}\cmd{\Arg{name}}\marg{arg spec}\marg{definition} \\
\cmd{NewDocumentEnvironment}\marg{name}\marg{arg spec}\marg{before}\marg{after}
\end{command}

相比 \cmd{newcommand} 和 \cmd{newenvironment}，\pkg{xparse} 通过 \marg{arg spec} 来指定参数的个数
和格式。基本的参数格式见表~\ref{tbl:xparse-arg-spec}。注意 \marg{arg spec} 中的空格可以忽略。

\begin{table}[htp]
\centering
\caption{\pkg{xparse} 参数格式}\label{tbl:xparse-arg-spec}
\begin{tabular}{cl}
 \toprule
 \textbf{参数格式} & \textbf{说明} \\
 \midrule
 \verb|m|               & 必选参数，由 \verb|{...}| 给出 \\
 \verb|o|               & 可选参数，由 \verb|[...]| 给出；未给出时返回 \verb|-NoValue-| 标记 \\
 \verb|O|\marg{default} & 可选参数，但在未给出时则返回默认值 \Arg{default} \\
 \verb|s|               & 可选的星号 \verb|*| \\
 \verb|+|               & 修饰后面的 \verb|m|、\verb|o| 等，表示允许在参数中分段 \\
 \bottomrule
\end{tabular}
\end{table}

不同输入值在解析后的结果可以见表~\ref{tbl:xparse-arg-input} 中的示例。

\begin{table}[htp]
\centering
\caption{\pkg{xparse} 参数示例}\label{tbl:xparse-arg-input}
\begin{tabular}{ccccc}
 \toprule
 \textbf{参数格式} & \textbf{输入值} & \verb|#1| & \verb|#2| & \verb|#3| \\
 \midrule
 \verb|m m|            & \verb|{foo}{bar}|  & \verb|foo|         & \verb|bar|       &                \\
 \verb|o m|            & \verb|{foo}|       & \verb|-NoValue-|   & \verb|foo|       &                \\
 \verb|o o m|          & \verb|[foo]{bar}|  & \verb|foo|         & \verb|-NoValue-| & \verb|bar|     \\
 \verb|m O{default}|   & \verb|{foo}|       & \verb|foo|         & \verb|default|   &                \\
 \verb|m O{default}|   & \verb|{foo}[bar]|  & \verb|foo|         & \verb|bar|       &                \\
 \verb|m O{default}|   & \verb|[bar]|       & 报错               &                  &                \\
 \verb|s o m|          & \verb|*[foo]{bar}| & \cmd{BooleanTrue}  & \verb|foo|       & \verb|bar|     \\
 \verb|s m O{default}| & \verb|{foo}|       & \cmd{BooleanFalse} & \verb|foo|       & \verb|default| \\
 \bottomrule
\end{tabular}
\end{table}

\cmdindex[xparse]{IfNoValueTF,IfNoValueT,IfNoValueF}
\verb|-NoValue-| 标记可以用 \cmd{IfNoValueTF} 等命令来判断：

\begin{command}
\cmd{IfNoValueTF}\marg{argument}\marg{true code}\marg{false code} \\
\cmd{IfNoValueT}\marg{argument}\marg{true code} \\
\cmd{IfNoValueF}\marg{argument}\marg{false code}
\end{command}

举例如下：

\begin{example}
% 百分号用于注释掉不必要的空格和换行符
\NewDocumentCommand\hello{om}
  {%
    \IfNoValueTF{#1}%
      {Hello, #2!}%
      {Hello, #1 and #2!}%
  }
\hello{Alice}
\hello[Bob]{Alice}
\end{example}

\cmdindex[xparse]{BooleanTrue,BooleanFalse,IfBooleanTF,IfBooleanT,IfBooleanF}
\cmd{BooleanTrue} 和 \cmd{BooleanFalse} 则可以用 \cmd{IfBooleanTF} 等命令来判断：

\begin{command}
\cmd{IfBooleanTF}\marg{argument}\marg{true code}\marg{false code} \\
\cmd{IfBooleanT}\marg{argument}\marg{true code} \\
\cmd{IfBooleanF}\marg{argument}\marg{false code}
\end{command}

举例如下：

\begin{example}
\NewDocumentCommand\hereis{sm}
  {Here is \IfBooleanTF{#1}{an}{a} #2.}
\hereis{banana}
\hereis*{apple}
\end{example}

需要注意的是，与命令不同，环境在定义时名字里面可以包含 \verb|*|：
\begin{verbatim}
\NewDocumentEnvironment {mytabular}  { o +m } {...} {...}
\NewDocumentEnvironment {mytabular*} { m o +m } {...} {...}
\end{verbatim}
用 \verb|s| 标记的 \verb|*| 则应该放在 \cmd{begin}\marg{env} 的后面：
\begin{verbatim}
\NewDocumentEnvironment { envstar } { s }
  {\IfBooleanTF {#1} {star} {no star}} {}
\begin{envstar}*
\end{envstar}
\end{verbatim}

\cmdindex[xparse]{RenewDocumentCommand,ProvideDocumentCommand,DeclareDocumentCommand}
\cmdindex[xparse]{RenewDocumentEnvironment,ProvideDocumentEnvironment,DeclareDocumentEnvironment}
与 \cmd{renewcommand}、\cmd{providecommand} 等命令类似，\pkg{xparse} 宏包也允许在命令或环境
已有定义时做出相应的处理，具体见表~\ref{tbl:xparse-renew}。

\begin{table}[htp]
\centering
\caption{\pkg{xparse} 提供的其他命令}\label{tbl:xparse-renew}
\begin{tabular}{cccc}
 \toprule
 \textbf{定义命令} & \textbf{定义环境} & \textbf{已定义} & \textbf{未定义} \\
 \midrule
 \cmd{NewDocumentCommand}     & \cmd{NewDocumentEnvironment}     & 报错       & 给出定义 \\
 \cmd{RenewDocumentCommand}   & \cmd{RenewDocumentEnvironment}   & 重新定义   & 报错     \\
 \cmd{ProvideDocumentCommand} & \cmd{ProvideDocumentEnvironment} & 什么也不做 & 给出定义 \\
 \cmd{DeclareDocumentCommand} & \cmd{DeclareDocumentEnvironment} & 重新定义   & 给出定义 \\
 \bottomrule
\end{tabular}
\end{table}

\section{编写自己的宏包和文档类}\label{sec:packages}

\subsection{编写简单的宏包}\label{subsec:provide-pkg}

\cmdindex{usepackage}
如果定义了很多新的环境和命令，文档的导言区将变得很长，在这种情况下，可以建立一个新的 \LaTeX{} 宏包
来存放所有你自己定义的命令和环境，然后在文档中使用 \cmd{use\-package} 命令来调用自定义的宏包。

\begin{sourcecode}[htp]
\begin{Verbatim}
% Demo Package by Tobias Oetiker
\ProvidesPackage{demopack}
\newcommand{\tnss}{The not so Short Introduction
                   to \LaTeXe}
\newcommand{\txsit}[1]{The \emph{#1} Short
                       Introduction to \LaTeXe}
\newenvironment{king}{\begin{quote}}{\end{quote}}
\end{Verbatim}
\caption{宏包的一个最简示例。}\label{code:package}
\end{sourcecode}

\cmdindex{ProvidesPackage}
写一个宏包的基本工作就是将原本在你的文档导言区里很长的内容拷贝到另一个文件中去，
 这个文件需要以 \texttt{.sty} 作扩展名。你还需要加入一个宏包专用的命令：
\begin{command}
\cmd{ProvidesPackage}\marg{package name}
\end{command}
这个命令应该放在你的宏包的最前面，并且一定要注意：\textbf{\Arg{package name} 需要和宏包的文件名一致。}
\cmd{Provides\-Package} 让 \LaTeX{} 记录宏包的名称，从而在 \cmd{usepackage} 命令再次调用同一个宏包的时候忽略之%
\footnote{但如果你以\emph{不同的选项}多次引入宏包，则有可能会引起错误，见附录 \ref{sec:errors}。}。
源代码 \ref{code:package} 给出了一个小的宏包示例，其中包含了我们之前定义的一些命令。

\subsection{在宏包中调用其它宏包}\label{subsec:require-pkg}

\cmdindex{RequirePackage}
如果你想进一步把各种宏包的功能汇总到一个文件里，而不是在文档的导言区罗列一大堆宏包的话，
\LaTeX{} 允许你在自己编写的宏包中调用其它宏包，命令为 \cmd{RequirePackage}，用法和 \cmd{usepackage}
一致：
\begin{command}
\cmd{RequirePackage}\oarg{options}\marg{package name}
\end{command}

\subsection{编写自己的文档类}\label{subsec:provide-cls}

\cmdindex{ProvidesClass}
当你更进一步，需要编写自己的文档类，如论文模板等，问题就稍稍麻烦了一些。首先，自己的文档类以 \texttt{.cls} 作扩展名，开头使用
\cmd{ProvidesClass} 命令：
\begin{command}
\cmd{ProvidesClass}\marg{class name}
\end{command}
当然了，\marg{class name}也需要和文档类的文件名一致。

\cmdindex{LoadClass}
但是有了上述命令和和你之前学到的 \cmd{newcommand} 等，还并不能完成一个文档类的编写，因为诸如 \cmd{chapter}、
\cmd{section} 等等许多常用的命令都是在文档类中定义的。事实上，许多时候我们只需要像调用宏包那样调用一个基本的文档类，
省去许多不必要的麻烦。在你的文档类中调用其它文档类的命令是 \cmd{LoadClass}，用法和 \cmd{documentclass} 十分相像：
\begin{command}
\cmd{LoadClass}\oarg{options}\marg{package name}
\end{command}

\section{计数器}\label{sec:counters}

\pinyinindex{jishuqi}{计数器}
我们早就见识到了 \LaTeX{} 对文档元素自动计数的能力：章节符号、列表、图表……它们都是依靠 \LaTeX{} 提供的计数器功能完成的。

\subsection{定义和修改计数器}\label{subsec:count-defs}

\cmdindex{newcounter}
定义一个计数器的方法为：
\begin{command}
\cmd{newcounter}\marg{counter name}\oarg{parent counter name}
\end{command}

\Arg{counter name} 为计数器的名称。计数器可以有上下级的关系，可选参数 \Arg{parent counter name} 定义为 \Arg{counter name} 的上级计数器。

\cmdindex{setcounter,addtocounter,stepcounter}
以下命令修改计数器的数值，\cmd{set\-counter} 将数值设为 \Arg{number}；\cmd{add\-to\-counter} 将数值加上 \Arg{number}；
\cmd{step\-counter} 将数值加一，并\textbf{将所有下级计数器归零}。
\begin{command}
\cmd{setcounter}\marg{counter name}\marg{number} \\
\cmd{addtocounter}\marg{counter name}\marg{number} \\
\cmd{stepcounter}\marg{counter name}
\end{command}

\subsection{计数器的输出格式}\label{subsec:count-value}

计数器 \Arg{counter} 的输出格式由 \cmd{the\Arg{counter}} 表示，如我们在 \ref{sec:pagestyle} 一节见过的 \cmd{thechapter} 等。
这个值默认以阿拉伯数字形式输出，如果想改成其它形式，需要重定义 \cmd{the\Arg{counter}}，如将 equation 计数器的格式定义为大写字母：
\begin{verbatim}
\renewcommand\theequation{\Alph{equation}}
\end{verbatim}

\cmdindex{arabic,alph,Alph,roman,Roman,fnsymbol}
命令 \cmd{Alph} 控制计数器 \Arg{counter} 的值以大写字母形式显示。下表列出所有可用于修改计数器格式的命令。
注意：这些命令\textbf{只能用于计数器，不能直接用于数字}，如 \cmd{roman}\marg*{1} 这样的命令会出错。
\begin{table}[htp]
\centering
\caption{计数器输出格式相关命令}\label{tbl:counter-commands}
\begin{tabular}{lp{22em}l}
 \hline
 \textbf{命令} & \textbf{样式} & \textbf{范围} \\
 \hline
 \cmd{arabic} & 阿拉伯数字（默认） & \\
 \cmd{alph}  & 小写字母 & 限 0--26 \\
 \cmd{Alph}  & 大写字母 & 限 0--26 \\
 \cmd{roman} & 小写罗马数字 & 限非负整数 \\
 \cmd{Roman} & 大写罗马数字 & 限非负整数 \\
 \cmd{fnsymbol} & 一系列符号，用于 \cmd{thanks} 命令生成的脚注 & 限 0--9 \\
 \hline
\end{tabular}
\begin{quotation}
\small
\makeatletter
注：\cmd{fnsymbol} 使用的符号顺次为：
\@fnsymbol{1} \@fnsymbol{2} \@fnsymbol{3}
\@fnsymbol{4} \@fnsymbol{5} \@fnsymbol{6}
\@fnsymbol{7} \@fnsymbol{8} \@fnsymbol{9}
\makeatother
\end{quotation}
\end{table}

计数器的输出格式还可以利用其它字符，甚至其它计数器的输出格式与之组合。如标准文档类里对 \cmd{sub\-section} 相关的计数器的输出格式的定义相当于：
\begin{verbatim}
\renewcommand\thesubsection{\thesection.\arabic{subsection}}
\end{verbatim}

\subsection{\LaTeX{} 中的计数器}\label{subsec:latex-counts}

\begin{itemize}
  \item 所有章节命令 \cmd{chapter}、\cmd{section} 等分别对应计数器 chapter、section 等等，而且有上下级的关系。
        而计数器 part 是独立的。
  \item 有序列表 \env{enumerate} 的各级计数器为 enumi, enumii, enumiii, enumiv，也有上下级的关系。
  \item 图表浮动体的计数器就是 table 和 figure；公式的计数器为 equation。
        这些计数器在 \cls{article} 文档类中是独立的，而在 \cls{report} 和 \cls{book} 中以 chapter 为上级计数器。
  \item 页码、脚注的计数器分别是 page 和 footnote。
\end{itemize}

我们可以利用前面介绍过的命令，修改计数器的样式以达到想要的效果，比如把页码修改成大写罗马数字，左右加横线，或是给脚注加上方括号：
\begin{verbatim}
\renewcommand\thepage{--~\Roman{page}~--}
\renewcommand\thefootnote{[\arabic{footnote}]}
\end{verbatim}

\cmdindex{pagenumbering}
我们在 \ref{subsec:basic-pagesyle} 小节中见过命令 \cmd{pagenum\-bering}。
它的内部机制是修改 page 计数器的格式 \cmd{thepage}，并将计数器的值重置为 1。

最后介绍两个有用的计数器：

\subsubsection{secnumdepth}

\index{secnumdepth@secnumdepth (\textit{计数器})}
\LaTeX{} 标准文档类对章节划分了层级：
\begin{itemize}
  \item 在 \cls{article} 文档类里 part 为 0，section 为 1，依此类推；
  \item 在 \cls{report} / \cls{book} 文档类里 part 为 -1，chapter 为 0，section 为 1，等等。
\end{itemize}

secnumdepth 计数器控制章节编号的深度，如果章节的层级大于 secnumdepth，那么章节的标题、在目录和页眉页脚的标题都不编号
（照常生成目录和页眉页脚），章节计数器也不计数。

可以用 \cmd{setcounter} 命令设置 secnumdepth 为较大的数使得层级比较深的章节也编号，
如设置为 4 令 \cmd{paragraph} 也编号；或者设置一个较小的数以取消编号，如设置为 -1 令 \cmd{chapter} 不编号。
后者是生成不编号的章节的一个妙招，免去了手动使用 \cmd{add\-contents\-line} 和 \cmd{mark\-both} 的麻烦。

secnumdepth 计数器在 \cls{article} 文档类里默认为 3（subsubsection 一级）；在 \cls{report} 和 \cls{book} 文档类里默认为 2（subsection 一级）。

\subsubsection{tocdepth}

\index{tocdepth@tocdepth (\textit{计数器})}
tocdepth 计数器控制目录的深度，如果章节的层级大于 tocdepth，那么章节将不会自动写入目录项。默认值同 secnumdepth。

\section{\LaTeX{} 可定制的一些命令和参数}\label{sec:latex-settings}

\LaTeX{} 事实上有相当一些可以定制的命令和参数，不过对于修改样式或者开发宏包来说，这些定制项还远远不够。

对于用户来讲，容易定制的是这一些项目：

\begin{itemize}
  \item 标题名称/前后缀等。表 \ref{tbl:latex-settings-names} 列出了标准文档类里可定制的项目，
  表中所有的 \LaTeX{} 命令都可以用 \cmd{re\-new\-comm\-and} 来修改。
  \item 长度。前文在叙述各种排版元素时已经介绍过一些，现归纳于表 \ref{tbl:latex-settings-lengths}。
  表中所有的长度命令可用 \cmd{setlength} 来修改。
  大多数控制页面尺寸的长度参数在图 \ref{fig:layouts} 给出，此处不再赘述。
\end{itemize}

\begin{table}[htp]
\centering
\caption{\LaTeX{} 可定制的标题名称/前后缀}\label{tbl:latex-settings-names}
\small
\begin{tabular}{llp{24em}}
 \hline
 \textbf{命令} & \textbf{默认值} & \textbf{含义} \\
 \hline
 \cmd{partname}       & Part            & \cmd{part} 命令生成的标题前缀 \\
 \cmd{chaptername}    & Chapter         & \cmd{chapter} 命令生成的标题前缀 \\
 \cmd{appendixname}   & Appendix        & 使用 \cmd{appendix} 命令生成的附录部分的章标题前缀 \\
 \cmd{abstractname}   & Abstract        & 摘要环境 \env{abstract} 的标题名称 \\[1ex]
 \cmd{contentsname}   & Contents        & \cmd{tableofcontents} 命令生成的目录标题 \\
 \cmd{listfigurename} & List of Figures & \cmd{listoffigures} 命令生成的插图目录标题 \\
 \cmd{listtablename}  & List of Tables  & \cmd{listoftables} 命令生成的表格目录标题 \\[1ex]
 \cmd{tablename}      & Table           & \env{table} 浮动体中 \cmd{caption} 命令生成的标题前缀 \\
 \cmd{figurename}     & Figure          & \env{figure} 浮动体中 \cmd{caption} 命令生成的标题前缀 \\[1ex]
 \cmd{refname}        & References      & \env{thebibliography} 环境或 \cmd{biblio\-graphy}
                                          命令生成的参考文献标题（\cls{article} 文档类） \\
 \cmd{bibname}        & Bibliography    & \env{thebibliography} 环境或 \cmd{biblio\-graphy}
                                          命令生成的参考文献标题（\cls{report} / \cls{book} 文档类）\\
 \cmd{indexname}      & Index           & \cmd{printindex} 命令生成的索引标题 \\
 \hline
\end{tabular}
\begin{quotation}\footnotesize%
注：形如“第 X 章”和“第 X 部分”的中文章节标题不能直接由修改本表的命令得到，需要使用 \pkg{titlesec} 等宏包定制。
如果使用 \pkg{ctex} 宏包或文档类，那么标题默认被修改成“第 X 章”和“第 X 部分”的形式，本表中的其它标题也修改为中文标题。详见 \pkg{ctex} 宏包的帮助文档。
\end{quotation}
\end{table}

\begin{table}[htp]
\centering
\caption{\LaTeX{} 可定制的长度参数}\label{tbl:latex-settings-lengths}
\small
\begin{tabular}{lll}
 \hline
 \textbf{命令} & \textbf{默认值} & \textbf{含义} \\
 \hline
 \cmd{fboxrule}     & 0.4pt & \cmd{fbox} 或 \cmd{framebox} 等带框盒子的线宽 \\
 \cmd{fboxsep}      & 3pt   & \cmd{fbox} 或 \cmd{framebox} 等带框盒子的内边距 \\[1ex]
 \cmd{arraycolsep}  & 5pt   & \env{array} 环境的表格项前后的间距\textsuperscript{注 1} \\
 \cmd{tabcolsep}    & 6pt   & \env{tabular} 环境的表格项前后的间距\textsuperscript{注 1} \\
 \cmd{arrayrulewidth}     &  0.4pt  & 表格线宽 \\
 \cmd{doublerulesep}      &  2pt    & 连续两根表格线之间的间距 \\[1ex]
 \cmd{abovecaptionskip}   &  10pt   & \cmd{caption} 命令上方的间距\textsuperscript{注 2} \\
 \cmd{belowcaptionskip}   &  0pt    & \cmd{caption} 命令下方的间距\textsuperscript{注 2} \\[1ex]
 \cmd{columnsep}          &  10pt   & 双栏排版下两栏的间距 \\
 \cmd{columnseprule}      &  0pt    & 双栏排版下两栏之间竖线的宽度 \\
 \hline
\end{tabular}
\begin{quotation}\footnotesize%
注 1：\cmd{arraycolsep} 和 \cmd{tabcolsep} 是每个表格项本身前后的间距（表格线前后无间距；\texttt{@} 列格式会消除与前后表格项的间距）。
两个表格项之间的间距相当于 2\cmd{arraycolsep} 或 2\cmd{tabcolsep}。\par
注 2：在默认设置下，\cmd{caption} 命令和位于它下方的图表之间无间距。宏包 \pkg{caption} 改善了这个问题。
\end{quotation}
\end{table}

\endinput
